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Análisis 



SHIFTAHOLIC 


Qué es Shiftaholic 

Aplicación web que permite gestionar los turnos 

Aplicación móvil que permite solicitar y recibir peticiones de cambio de turnos 


Para quién es Shiftaholic 

Empresas con gestión de turnos compleja en la que los cambios de turno 
son habituales. Colectivos como médicos, enfermeros, controladores... 
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Tecnologías 


Android (Android Studio) 

JavaEE (Eclipse) 

Web (Eclipse) 

• Realm 

• Jersey 

• Bootstrap 

• EventBus 


• JQuery 

• Retrofit 
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Casos de uso 
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Roles 


• Administrador 

• Recursos Humanos 


• Usuario 
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Administrador 


Shiftaholic 

£ 

admin 

Usuarios 

ID 

DNI 

Nombre 

Apellido 

2° Apellido 

Email 

Activo 




Ver 

malfs 

385619241 

Mónica 

Alférez 

Siruella 

muchomorgJhotmail .com 

✓ 

/ 

ffl 

e a 

Registro 

málv 

32562563Y 

Miguel 

Álvarez 


miguelalig)gmail.com 

✓ 

/ 

d 

í? A 

Archivos 

rarrm 

2566315p 

Ricard 

Arroyo 

Martín 

rarrmaiSgmail.com 

✓ 

/ 

ffi 

§ A 


jcocs 

38561923P 

Jessica 

Cocerá 

Soria 

mantisbboySgmail com 

✓ 

/ 

d 

A 


mcocl 

48007284M 

Marco A. 

Cocerá 

López 

marcoacoloSgmail.com 

✓ 

/ 

s 

e a 
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Recursos Humanos 
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Recursos Humanos 


Shiftaholic 


Usuarios 

ID 

DNI 

Nombre 

Apellido 

2° Apellido 

Grupos 

malfs 

385619241 

Ménica 

Alférez 

Siruella 

Ver grupos 

málv 

32562563Y 

Miguel 

Álvarez 


Gestionar turnos 

rarrm 

2566315p 

Ricard 

Arroyo 

Martín 


jcocs 

38561923P 

Jessica 

Cocerá 

Soria 

Ver peticiones 







mcocl 

48007284M 

Marco A. 

Cocerá 

López 







Configurar turnos 

ccrul 

38561924F 

Cristian 

Cruz 

Lisonja 







dcrum 

38561924Z 

Daniel 

Cruz 

Morales 


Email 

muchomotgíhotmail.com 
miguelalig)gmail com 
rarrmaig)gmail .com 
mantisbboyigJgmail com 
marcoacoloia)gmail.com 
crispelkgJgmail com 
dani.cruz.moralesia)gmail.com 



Subir foto 
Gn salir C* 


|_M; 


anana ▼ 


Indefinido ▼ 


Indefinido T 


Indefinido T 


Indefinido 


Indefinido T 


Indefinido ▼ 
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Usuario (trabajador) 


IOLIC 




SHIFTV 


Usuario (trabajador) 


DLK 



% 

* 



Mónica Alférez 
malfs 


& 


±_ 

* 


Mi Horario 

Ver ofertas 
Peticiones enviadas 
Peticiones recibidas 


Otros 

Salir 
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Funcionamiento 
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Descripción de la Base de datos 
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Base de datos - users 


Columna | Tipo | Modificadores 










user id 


1 

character 

varying 

1 

not 

nuil 

external 

id 

1 

character 

varying 

1 

not 

nuil 

ñame 


1 

character 

varying 

1 

not 

nuil 

surnamel 


1 

character 

varying 

1 

not 

nuil 

surname2 


1 

character 

varying 

1 



email 


1 

character 

varying 

1 

not 

nuil 

creation 

time 

1 

bigint 


1 

not 

nuil 

removal 

time 

1 

bigint 


1 
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Base de datos - shift_type 


Columna 


Tipo 


Modificadores 


- + - 


- + - 


id 

1 

integer 

1 

not 

nuil 

ñame 

1 

character varying 

1 

not 

nuil 

real hours 

1 

numer ic (4,2) 

1 

not 

nuil 

computed hours 

1 

numeric (4,2) 

1 

not 

nuil 

ini time 

1 

character varying (5) 

1 

not 

nuil 

end time 

1 

character varying (5) 

1 

not 

nuil 

shift generator 

1 

integer [ ] 

1 

not 

nuil 

week day 

1 

integer 

1 

not 

nuil 

next assignement 

1 

bigint 

1 
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Base de datos - shift 


Columna | Tipo | Modificadores 

1 1 

id 

integer 

not nuil 

user id 

character varying 

not nuil 

shift type 

integer 

not nuil 

ini time 

character varying (5) 

not nuil 

end time 

character varying (5) 

not nuil 

computed hours 

numeric (4,2) 

not nuil 

day 

bigint 

not nuil 

day type 

character varying 

not nuil 

worked 

boolean 

not nuil 

changeable 

boolean 


pending 

boolean 


request shift id 

integer 


last modification 

bigint 
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Base de datos - request_shift 


Columna | Tipo | Modificadores 

1 1 

id 

integer 

not nuil 

own shift 

integer 

not nuil 

other shift 

integer 

not nuil 

own free shift 

integer 


other free shift 

integer 


requester user 

character varying 

not nuil 

requested user 

character varying 

not nuil 

rrhh user 

character varying 


State 

integer 

not nuil 

reason 

text 


last modification 

bigint 
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Descripción del Servidor 
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Funcionamiento del servidor 

web 




Funcionamiento del servidor - web 


main.jsp 


AHOLIC 
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Funcionamiento del servidor - web 


Solicitud del cliente 

function fetchData(contentid, targetid) { 

var set; 

if (targetid) { 

set = function (data) { 

$('#' + targetid) . html (data) ; 

>; 

} 

var req = new ContentRequest (contentid) ; 

req.access=user Access; 
var par = JSON. stringif y ( req) ; 
console.log (req) ; 
consolé . log (par ) ; 

$.ajax({ 

url : CONTENTURL, 
type : 'POST', 
data : par, 

contentType : " application/j son; charset=utf - 8" , 
dataType : "html", 
success : set 

>); 


function ContentRequest (contentid) { 
this.content = contentid; 

this.token = sessionStorage . getltem( " token" ) ; 

} 


} 




SHIFTAHOLIC 


Funcionamiento del servidor - web 


Procesado del servidor 

@WebServlet (ñame = M app", urlPatterns = { u /web/app" }) 
public class PagesServer extends HttpServlet { 

prívate static final long serialVersionUID = 1L ; 
prívate static final String URL_BASE = "/WEB-INF/j sp/“ ; 

prívate void process (HttpServletRequest request, HttpServletResponse response) 
throws ServletException , IOException { 

// Cargamos la request que viene como json 

ContentRequest contentRequest = new Gson ( ) . f romJson ( new InputStreamReader( request .getInputStream( )) , 
ContentRequest . class) ; 

if (contentRequest == nuil || !isValidRequest(contentRequest, request)) { 

// TODO que hacer cuando no es valida. 

return ; 

} 


dispatchFragment (contentRequest , request, response); 


} 
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Funcionamiento dei servidor - web 

Procesado del servidor 


private void dispatcbFragment ( ContentRequest contentRequest r HttpServletRequest request, 
HttpServletResponse response) throws ServletException , IQException { 

St ring content = contentRequest . getContent f ) ; 

if ( content . equals ( "app" ) && contentRequest . getAccessí ). equalsIgnoreCasef "admin f ‘ ) J { 
request . setAttribute ( "access" , r 'adrnin" ) ; 

} 

request . getRequestDispatcher[ÜRL_EflS£ + “f ragments/"’ + content + n .jsp") 

. forward ( request , response); 


} 
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Funcionamiento del servidor 

rest 
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Funcionamiento del servidor - rest 



Gestora 


DAO 
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Funcionamiento del servidor - rest 

function doRestPost ( req, url, callback, errorCallback) { 

var par = JSON. stnngif y [ req) ; 
consolé . log ( req) ; 

$. ai ax [{ 

url : REST URL + url, 
type : 1 POST Í , 
data : par, 
async : true, 

conteritType : 11 appli catión/ j son; charset=utf - 8" , 
dataType : 11 1 son", 
success : callback, 
error : errorCallback 

>); 

> 


Solicitud del 

cliente 

web 
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Funcionamiento del servidor - rest 


public void dol_ogin(String username, String password) { 

RestRequest request = new RestRequest() ; 
request.getParams() .put("us£rid", username) ; 
request. getParams() .put("password" , password) ; 

userServices.postLogin(request) .enqueue(new Callback<LoginResponse>() { 

@Override 

public void onResponse(Call<LoginResponse> cali, Response<LoginResponse> response) { 

Log.d("doLogin" , "response ok - " + response. body() ) ; 

EventBus.getDefoult() . post (response. body() ) ; 

} 

@Override 

public void onFailure(Call<LoginResponse> cali, Throwable t) { 

Log.e("doLogin" , "response ko - " + cali. request() .url() , t) ; 
postError(serverError) ; 

1 ); 

} 


Solicitud del 

cliente 

android 
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Funcionamiento del servidor - rest 


@Path [ "login" ) 

(3P0ST 

(^Consumes [MediaType. APPL ICA TIONJSON ) 

^Produces [MediaType .APPLICATION_JSON) 
public LoginResponse login [RestRequest request) { 

LoginResponse response = new LoginResponse U ; 

Map<String, String> params = request , getParamsí ) ; 

Login login = manager . doLogin f params . get [ "userid" ) , params . get í "password" )) ; 

String errorCode = login . getErrorCode [) ; 

response . set ErrorCode (errorCode) ; 

if ( errorCode. equals í Er r orCoáes .ERROR_CODE_NO_ERROR ) ) { 
response . setLastTime (login . getLastLogin ( ) ) ; 
response . setToken [login . getToken [ ) ) ; 
response . set User Id [login . getUserld [ ) ) ; 
response . setName [login . getName [ ) ) ; 
response . set Su mame [login . get Su mame [ ) ) ; 

} 

return response; 


Procesado 

del 

Servidor 
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Funcionamiento del servidor 

archivos csv 
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Funcionamiento del servidor - archivos csv (subida) 



Gestora 

archivos 



DAO 

usuario 
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Funcionamiento del servidor - archivos csv (bajada) 



Gestora 

archivos 


BD 
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Funcionamiento del servidor - archivos csv (bajada) 

@GET 

@Path( ”download/{type}/{bundle}/{filename}") 

(aproduces (MediaType .APPLICATIONOCTETSTREAM) 

public Response downloadCSVFile(@QueryParam("token") String token, 
@PathParam( "type" ) String type, 

@PathParam( "bundle" ) String bundle, 

@PathParam( "filename" ) String ñame) { 
if ( !userManager.validToken(token) ) { 

return Response. status (Response. Status. UNAUTHORIZED) .buildO ; 

} 

FileType ftype = nuil; 
switch (type) { 
case " registration" : 

ftype = FileTy pe. RE GIS TRATIONFILE; 
break; 

default: 

return Response. status (Response. Status. BADREQUEST) .buildO ; 


> 

DatabaseBinary file = fileManager.getFile(bundle, ñame, ftype); 
if (file == nuil) { 

return Response. status (Response. Status. NOTFOUND) .buildO ; 

} else { 

return Response. ok ( f ile. getSt ream ( ) ) . build( ) ; 

} 
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Instalación 
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Requisitos del Servidor 


• Tomcatv8 


• PostgreSQL v9.5 
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Demo 


http://80.28. 157.253:81 81 /shiftaholic/web/app 



